home *** CD-ROM | disk | FTP | other *** search
/ Monster Media 1996 #15 / Monster Media Number 15 (Monster Media)(July 1996).ISO / bbs_util / bsrc_260.zip / SRC.ZIP / RECVSYNC.C < prev    next >
C/C++ Source or Header  |  1996-03-23  |  25KB  |  721 lines

  1. /*--------------------------------------------------------------------------*/
  2. /*                                                                          */
  3. /*                                                                          */
  4. /*      ------------         Bit-Bucket Software, Co.                       */
  5. /*      \ 10001101 /         Writers and Distributors of                    */
  6. /*       \ 011110 /          Freely Available<tm> Software.                 */
  7. /*        \ 1011 /                                                          */
  8. /*         ------                                                           */
  9. /*                                                                          */
  10. /*              (C) Copyright 1987-96, Bit Bucket Software Co.              */
  11. /*                                                                          */
  12. /*               This module was written by Vince Perriello                 */
  13. /*                                                                          */
  14. /*          FidoNet(R) Mail Session Called System Synchronization           */
  15. /*                                                                          */
  16. /*                                                                          */
  17. /*    For complete  details  of the licensing restrictions, please refer    */
  18. /*    to the License  agreement,  which  is published in its entirety in    */
  19. /*    the MAKEFILE and BT.C, and also contained in the file LICENSE.260.    */
  20. /*                                                                          */
  21. /*    USE  OF THIS FILE IS SUBJECT TO THE  RESTRICTIONS CONTAINED IN THE    */
  22. /*    BINKLEYTERM  LICENSING  AGREEMENT.  IF YOU DO NOT FIND THE TEXT OF    */
  23. /*    THIS  AGREEMENT IN ANY OF THE  AFOREMENTIONED FILES,  OR IF YOU DO    */
  24. /*    NOT HAVE THESE FILES,  YOU  SHOULD  IMMEDIATELY CONTACT BIT BUCKET    */
  25. /*    SOFTWARE CO.  AT ONE OF THE  ADDRESSES  LISTED BELOW.  IN NO EVENT    */
  26. /*    SHOULD YOU  PROCEED TO USE THIS FILE  WITHOUT HAVING  ACCEPTED THE    */
  27. /*    TERMS  OF  THE  BINKLEYTERM  LICENSING  AGREEMENT,  OR  SUCH OTHER    */
  28. /*    AGREEMENT AS YOU ARE ABLE TO REACH WITH BIT BUCKET SOFTWARE, CO.      */
  29. /*                                                                          */
  30. /*                                                                          */
  31. /* You can contact Bit Bucket Software Co. at any one of the following      */
  32. /* addresses:                                                               */
  33. /*                                                                          */
  34. /* Bit Bucket Software Co.        FidoNet  1:104/501, 1:343/491             */
  35. /* P.O. Box 460398                AlterNet 7:42/1491                        */
  36. /* Aurora, CO 80046               BBS-Net  86:2030/1                        */
  37. /*                                Internet f491.n343.z1.fidonet.org         */
  38. /*                                                                          */
  39. /* Please feel free to contact us at any time to share your comments about  */
  40. /* our software and/or licensing policies.                                  */
  41. /*                                                                          */
  42. /*--------------------------------------------------------------------------*/
  43.  
  44. /* Include this file before any other includes or defines! */
  45.  
  46. #include "includes.h"
  47.  
  48. /*
  49.     Receiver Synchronization state table.
  50.  
  51.     This logic is used by the called system. It will determine the
  52.     type of mail transfer which can be used in communicating with
  53.     the calling system.
  54.  
  55.     This stuff was originally copied from a file sent to us by TJ.
  56.     (Of course, then we hacked the heck out of it!!!)
  57.  
  58.     Thanks, Tom.
  59.  
  60.  
  61.  .-----+----------+-------------------------+-------------------------+-----.
  62.  | RS0 | SyncInit |                         | Start 5 second idle tmr | RS1 |
  63.  |-----+----------+-------------------------+-------------------------+-----|
  64.  | RS1 | IdleWait | 1. 5 sec tmr expired    | Take the initiative     | RS2 |
  65.  |     |          +-------------------------+-------------------------+-----|
  66.  |     |          | 2. Carrier lost         | Session aborted         | exit|
  67.  |     |          +-------------------------+-------------------------+-----|
  68.  |     |          | 3. Peek = YOOHOO        | Looks like a live WaZOO | RS3 |
  69.  |     |          +-------------------------+-------------------------+-----|
  70.  |     |          | 4. Peek = TSYNC         | Live FSC001, we think   | RS3 |
  71.  |     |          +-------------------------+-------------------------+-----|
  72.  |     |          | 5. Peek = CR, LF, space | He looks alive          | RS2 |
  73.  |     |          +-------------------------+-------------------------+-----|
  74.  |     |          | 6. Peek = '*'           | Looks like a live EMSI  | RS3 |
  75.  |     |          +-------------------------+-------------------------+-----|
  76.  |     |          | 7. Other character      | Eat it                  | RS1 |
  77.  |-----+----------+-------------------------+-------------------------+-----|
  78.  | RS2 |SendBanner| 1. Error returned       | Session aborted         | exit|
  79.  |     |          +-------------------------+-------------------------+-----|
  80.  |     |          | 2. Banner sent OK       |                         | RS3 |
  81.  |-----+----------+-------------------------+-------------------------+-----|
  82.  | RS3 |RecvInit  |                         | Init ext-mail scan ptrs | RS4 |
  83.  |     |          |                         | Start 20 sec timer      |     |
  84.  |     |          |                         | Init 10 sec timer       |     |
  85.  |-----+----------+-------------------------+-------------------------+-----|
  86.  | RS4 |SendSync  | 1. Error returned       | Session aborted         | exit|
  87.  |     |(xmit sync+-------------------------+-------------------------+-----|
  88.  |     |string)   | 2. String sent OK       | Watch for sender sync   | RS5 |
  89.  |-----+----------+-------------------------+-------------------------+-----|
  90.  | RS5 | WaitSync | 1. Carrier lost         | Session aborted         | exit|
  91.  |     |          +-------------------------+-------------------------+-----|
  92.  |     |          | 2. YOOHOO received      | WaZOO session selected  | exit|
  93.  |     |          +-------------------------+-------------------------+-----|
  94.  |     |          | 3. TSYNC received       | probable FSC001         | RS6 |
  95.  |     |          +-------------------------+-------------------------+-----|
  96.  |     |          | 4. CR received          | Still sync'ing          | RS4 |
  97.  |     |          +-------------------------+-------------------------+-----|
  98.  |     |          | 5. ESC received         | BBS session             | exit|
  99.  |     |          +-------------------------+-------------------------+-----|
  100.  |     |          | 6. '*' received         | probable EMSI           | RS8 |
  101.  |     |          +-------------------------+-------------------------+-----|
  102.  |     |          | 7. Other character rcvd | check ext-mail string   | RS7 |
  103.  |     |          +-------------------------+-------------------------+-----|
  104.  |     |          | 8. 10 sec timer elapsed | FSC001 protocol selected| exit|
  105.  |     |          +-------------------------+-------------------------+-----|
  106.  |     |          | 9. 20 sec timer elapsed | Not a mail session      | exit|
  107.  |-----+----------+-------------------------+-------------------------+-----|
  108.  | RS6 | TsyncTmr | 1. Timer not running    | Start 10 second timer   | RS5 |
  109.  |     |          |                         | Reset 20 sec timer      |     |
  110.  |     |          +-------------------------+-------------------------+-----|
  111.  |     |          | 2. Timer running        | Two TSYNCS = FTS-0001   | exit|
  112.  |-----+----------+-------------------------+-------------------------+-----|
  113.  | RS7 | ExtMail  | 1. String matched       | External mail selected  | exit|
  114.  |     |          +-------------------------+-------------------------+-----|
  115.  |     |          | 2. Checked all strings  | Get next input character| RS5 |
  116.  |-----+----------+-------------------------+-------------------------+-----|
  117.  | RS8 | EMSI     | 1. String matched       | Call EMSI handshake     | exit|
  118.  |     |          +-------------------------+-------------------------+-----|
  119.  |     |          | 2. No match             | Get next input character| RS5 |
  120.  `-----+----------+-------------------------+-------------------------+-----'
  121.  
  122. */
  123.  
  124. /*
  125.  * Data structure used by all RecvSync state machine functions.
  126.  * Contains all data which needs to be passed between various states.
  127.  *
  128.  */
  129.  
  130. typedef struct
  131. {
  132.     long control;                /* Must always start with a long!   */
  133.     long Idle_Timer;            /* 5 second initial idle timer      */
  134.     long BBS_Timer;                /* 20 second Timer for BBS user     */
  135.     char *ExtMailScan[16];        /* Scan pointer for External Mail   */
  136.     long TSYNC_Timer;            /* 10 second TSYNC timer            */
  137.     char *emsi_ptr;                /* pointer for EMSI_INQ             */
  138.     char *iemsi_ptr;            /* pointer for EMSI_CLI             */
  139.     int emsi_flag;
  140.     int result;                    /* Result we want to send out       */
  141. } RSARGS, *RSARGSP;
  142.  
  143. int RSSyncInit (RSARGSP);        /* Called by state machine at start */
  144. int RSExit (RSARGSP);            /* Called by state machine at end   */
  145. int RSIdleWait (RSARGSP);        /* RS1 state processing function    */
  146. int RSSendBannr (RSARGSP);        /* RS2 state processing function    */
  147. int RSRecvInit (RSARGSP);        /* RS3 state processing function    */
  148. int RSSendSync (RSARGSP);        /* RS4 state processing function    */
  149. int RSWaitSync (RSARGSP);        /* RS5 state processing function    */
  150. int RSTsyncTmr (RSARGSP);        /* RS6 state processing function    */
  151. int RSExtMail (RSARGSP);        /* RS7 state processing function    */
  152. int RSEMSIChk (RSARGSP);        /* RS8 state processing function    */
  153.  
  154. #define RS0    0                /* Reserved value of 0 for init     */
  155. #define RSexit 0                /* Slot 1 is exit, but called by 0  */
  156. #define RS1    2                /* First "user" slot is 2.          */
  157. #define RS2    3                /* After that, it all maps n : n+1  */
  158. #define RS3    4
  159. #define RS4    5
  160. #define RS5    6
  161. #define RS6    7
  162. #define RS7    8
  163. #define RS8    9
  164.  
  165. typedef struct
  166. {
  167.     char *state_name;
  168.     int (*state_func) (RSARGSP);
  169. } RSTATES, *RSTATEP;
  170.  
  171. RSTATES Recv_Sync[] =
  172. {                                /* Table used by state machine      */
  173.     {"RSSyncInit", RSSyncInit},    /* And referred to by 'SSn' defines */
  174.     {"RSExit", RSExit},            /* listed above ...                 */
  175.     {"RSIdleWait", RSIdleWait},
  176.     {"RSSendBannr", RSSendBannr},
  177.     {"RSRecvInit", RSRecvInit},
  178.     {"RSSendSync", RSSendSync},
  179.     {"RSWaitSync", RSWaitSync},
  180.     {"RSTsyncTmr", RSTsyncTmr},
  181.     {"RSExtMail", RSExtMail},
  182.     {"RSEMSIChk", RSEMSIChk},
  183. };
  184.  
  185. /*
  186.  * CalledRecvSync
  187.  * Determine whether the calling system is an FTS-0001 mailer, an FTS-0006
  188.  * mailer, an external mailer, or a human caller. Use the general state
  189.  * machine driver.
  190.  *
  191.  * This is the only external entry point into this module.
  192.  *
  193.  */
  194.  
  195. int 
  196. CalledRecvSync (void)
  197. {
  198.     RSARGS args;
  199.     int res;
  200.  
  201.     args.result = 0;
  202.  
  203.     res = state_machine ((STATEP) Recv_Sync, &args, 2);
  204.     return (res);
  205. }
  206.  
  207. /*
  208.  * This routine is called by the state machine when the 'RSexit'
  209.  * state is seen. Its return value is what the state machine
  210.  * will return to its caller as the result of the function.
  211.  *
  212.  */
  213.  
  214. int 
  215. RSExit (RSARGSP args)
  216. {
  217.     return (args->result);
  218. }
  219.  
  220. /*
  221.  
  222.  .-----+----------+-------------------------+-------------------------+-----.
  223.  | RS0 | SyncInit |                         | Start 2 second idle tmr | RS1 |
  224.  `-----+----------+-------------------------+-------------------------+-----'
  225.  
  226.  */
  227.  
  228. int 
  229. RSSyncInit (RSARGSP args)
  230. {
  231.     char buff[40];
  232.  
  233.     if (!no_EMSI_Session)
  234.     {
  235.         sprintf (buff, "%s\r              \r", emsistr[EMSI_REQ]);
  236.         if (!SendBanner ((char far *) buff))
  237.         {
  238.             args->result = 0;
  239.             return (RSexit);
  240.         }
  241.     }
  242.     args->Idle_Timer = timerset (200);
  243.     return (RS1);
  244. }
  245.  
  246. /*
  247.  
  248.  .-----+----------+-------------------------+-------------------------+-----.
  249.  | RS1 | IdleWait | 1. 2 sec tmr expired    | Take the initiative     | RS2 |
  250.  |     |          +-------------------------+-------------------------+-----|
  251.  |     |          | 2. Carrier lost         | Session aborted         | exit|
  252.  |     |          +-------------------------+-------------------------+-----|
  253.  |     |          | 3. Peek = YOOHOO        | Looks like a live WaZOO | RS3 |
  254.  |     |          +-------------------------+-------------------------+-----|
  255.  |     |          | 4. Peek = TSYNC         | Live FSC001, we think   | RS3 |
  256.  |     |          +-------------------------+-------------------------+-----|
  257.  |     |          | 5. Peek = CR, LF, space | He looks alive          | RS2 |
  258.  |     |          +-------------------------+-------------------------+-----|
  259.  |     |          | 6. Peek = '*'           | Looks like a live EMSI  | RS3 |
  260.  |     |          +-------------------------+-------------------------+-----|
  261.  |     |          | 7. Other character      | Eat it                  | RS1 |
  262.  `-----+----------+-------------------------+-------------------------+-----'
  263.  
  264. */
  265.  
  266. int 
  267. RSIdleWait (RSARGSP args)
  268. {
  269.     short i;
  270.     int ret;
  271.  
  272.     while (!timeup (args->Idle_Timer))
  273.     {
  274.  
  275.         if (!CARRIER)
  276.         {
  277.             args->result = 0;
  278.             return (RSexit);
  279.         }
  280.  
  281.         if ((i = PEEKBYTE ()) == -1)
  282.         {
  283.             time_release ();        /*PLF Sun  12-01-1991  04:19:36 */
  284.             continue;
  285.         }
  286.  
  287.         switch (i & 0xff)
  288.         {
  289.  
  290.         case TSYNC:
  291.         case YOOHOO:
  292.             ret = RS3;                /* Fast lane, no banner */
  293.             break;
  294.  
  295.         case CR:
  296.         case LV:
  297.         case ' ':
  298.         case ENQ:
  299.         case ESC:
  300.             ret = RS2;                /* He's alive, do banner*/
  301.             break;
  302.  
  303.         case '*':
  304.             if (!no_EMSI_Session)
  305.             {
  306.                 ret = RS3;            /* Fast lane, no banner */
  307.                 break;
  308.             }
  309.  
  310.             /* Fall through if not EMSI */
  311.  
  312.         default:
  313.             (void) TIMED_READ (0);    /* eat it up            */
  314.             ret = RS1;
  315.             break;
  316.         }
  317.         return (ret);
  318.  
  319.     }
  320.     return (RS2);                    /* Idle for 5 seconds   */
  321. }
  322.  
  323. /*
  324.  
  325.  .-----+----------+-------------------------+-------------------------+-----.
  326.  | RS2 |SendBanner| 1. Error returned       | Session aborted         | exit|
  327.  |     |          +-------------------------+-------------------------+-----|
  328.  |     |          | 2. Banner sent OK       |                         | RS3 |
  329.  `-----+----------+-------------------------+-------------------------+-----'
  330.  
  331.  */
  332.  
  333. int 
  334. RSSendBannr (RSARGSP args)
  335. {
  336.     char buff[128];
  337.  
  338.     if (!no_EMSI_Session && (PEEKBYTE () != '*'))
  339.     {
  340.         sprintf (buff, "%s\r", emsistr[EMSI_REQ]);
  341.         if (!SendBanner ((char far *) buff))
  342.         {
  343.             args->result = 0;
  344.             return (RSexit);
  345.         }
  346.     }
  347.  
  348.     (void) sprintf (buff, MSG_TXT (M_ADDRESS),
  349.         Full_Addr_Str (&my_addr),
  350.         ANNOUNCE,
  351.         ((serial == -1) ? MSG_TXT (M_UNREGISTERED) : " "));
  352.  
  353.     if (!SendBanner ((char far *) buff))
  354.     {
  355.         args->result = 0;
  356.         return (RSexit);
  357.     }
  358.  
  359.     if (PEEKBYTE () != '*')
  360.     {
  361.  
  362.         if (BBSbanner != NULL)
  363.         {                            /* If BBS name spec'ed, */
  364.             SENDBYTE ('\r');        /* make sure user gets  */
  365.             SENDBYTE ('\n');        /* a full cr/lf...      */
  366.  
  367.             if (!SendBanner ((char far *) BBSbanner))
  368.             {
  369.                 args->result = 0;
  370.                 return (RSexit);
  371.             }
  372.         }
  373.         SENDBYTE ('\r');            /* make sure user gets  */
  374.         SENDBYTE ('\n');            /* a full cr/lf...      */
  375.     }
  376.  
  377.     if (!mail_only && (PEEKBYTE () != '*'))
  378.     {
  379.         if (!SendBanner (BBSwelcome))
  380.         {
  381.             args->result = 0;
  382.             return (RSexit);
  383.         }
  384.     }
  385.  
  386.     return (RS3);
  387. }
  388.  
  389. /*
  390.  
  391.  .-----+----------+-------------------------+-------------------------+-----.
  392.  | RS3 |RecvInit  |                         | Init ext-mail scan ptrs | RS4 |
  393.  |     |          |                         | Start 20 sec timer      |     |
  394.  |     |          |                         | Init 10 sec timer       |     |
  395.  `-----+----------+-------------------------+-------------------------+-----'
  396.  
  397.  */
  398.  
  399. int 
  400. RSRecvInit (RSARGSP args)
  401. {
  402.     register int k;
  403.  
  404.     for (k = 0; k < num_ext_mail; k++)
  405.     {
  406.         args->ExtMailScan[k] = ext_mail_string[k];    /* UUCP handshake       */
  407.     }
  408.  
  409.     args->emsi_ptr = emsistr[EMSI_INQ];
  410.     args->iemsi_ptr = emsistr[EMSI_CLI];
  411.     args->emsi_flag = FALSE;
  412.  
  413.     args->BBS_Timer = timerset (BBStimeout);    /* 20 second timeout    */
  414.     args->TSYNC_Timer = 0L;                        /* Initialize for test  */
  415.  
  416.     if (!no_EMSI_Session && (PEEKBYTE () == '*'))
  417.         return (RS5);
  418.     else
  419.         return (RS4);
  420. }
  421.  
  422. /*
  423.  
  424.  .-----+----------+-------------------------+-------------------------+-----.
  425.  | RS4 |SendSync  | 1. Error returned       | Session aborted         | exit|
  426.  |     |(xmit sync+-------------------------+-------------------------+-----|
  427.  |     |string)   | 2. String sent OK       | Watch for sender sync   | RS5 |
  428.  `-----+----------+-------------------------+-------------------------+-----'
  429.  
  430.  */
  431.  
  432. int 
  433. RSSendSync (RSARGSP args)
  434. {
  435.     char far *c;
  436.     char buff[128];
  437.  
  438.     if (!no_EMSI_Session)
  439.     {
  440.         /* If we already have a '*', just go handle it. */
  441.  
  442.         if (PEEKBYTE () == '*')
  443.             return (RS5);
  444.  
  445.         sprintf (buff, "%s\r", emsistr[EMSI_REQ]);
  446.         if (!SendBanner ((char far *) buff))
  447.         {
  448.             args->result = 0;
  449.             return (RSexit);
  450.         }
  451.     }
  452.  
  453.     if (mail_only)
  454.     {                                /* If no BBS allowed,   */
  455.         c = (char far *) noBBS;        /* tell human to git    */
  456.     }
  457.     else
  458.     {
  459.         c = (char far *) BBSesc;    /* or hit ESC for BBS   */
  460.     }
  461.  
  462.     if (!SendBanner (c))
  463.     {
  464.         args->result = 0;
  465.         return (RSexit);
  466.     }
  467.  
  468.     return (RS5);
  469. }
  470.  
  471. /*
  472.  
  473.  .-----+----------+-------------------------+-------------------------+-----.
  474.  | RS5 | WaitSync | 1. Carrier lost         | Session aborted         | exit|
  475.  |     |          +-------------------------+-------------------------+-----|
  476.  |     |          | 2. YOOHOO received      | WaZOO session selected  | exit|
  477.  |     |          +-------------------------+-------------------------+-----|
  478.  |     |          | 3. TSYNC received       | probable FSC001         | RS6 |
  479.  |     |          +-------------------------+-------------------------+-----|
  480.  |     |          | 4. CR received          | Still sync'ing          | RS4 |
  481.  |     |          +-------------------------+-------------------------+-----|
  482.  |     |          | 5. ESC received         | BBS session             | exit|
  483.  |     |          +-------------------------+-------------------------+-----|
  484.  |     |          | 6. '*' received         | probable EMSI           | RS8 |
  485.  |     |          +-------------------------+-------------------------+-----|
  486.  |     |          | 7. Other character rcvd | check ext-mail string   | RS7 |
  487.  |     |          +-------------------------+-------------------------+-----|
  488.  |     |          | 8. 10 sec timer elapsed | FSC001 protocol selected| exit|
  489.  |     |          +-------------------------+-------------------------+-----|
  490.  |     |          | 9. 20 sec timer elapsed | Not a mail session      | exit|
  491.  `-----+----------+-------------------------+-------------------------+-----'
  492.  
  493.  */
  494.  
  495. int 
  496. RSWaitSync (RSARGSP args)
  497. {
  498.     short i;
  499.     int ret = -1;
  500.  
  501.     while (CARRIER)
  502.     {
  503.  
  504.         if (got_ESC ())
  505.         {                                /* Manual abort?        */
  506.             LOWER_DTR ();                /* Yes, drop carrier    */
  507.             timer (10);
  508.             args->result = 0;
  509.             return (RSexit);
  510.         }
  511.  
  512.         /* Note: the following line is a NON-DESTRUCTIVE READ! */
  513.  
  514.         switch ((i = (short) PEEKBYTE ()) & 0xff)
  515.         {
  516.         case YOOHOO:                    /* Looks like a WaZOO   */
  517.  
  518.             if (no_WaZOO_Session)
  519.             {                            /* If we're not WaZOO,  */
  520.                 (void) TIMED_READ (0);    /* Eat the YooHoo       */
  521.                 continue;
  522.             }
  523.  
  524.             CLEAR_OUTBOUND ();            /* End noisy banner now */
  525.             args->result = 3;            /* WaZOO session        */
  526.             ret = RSexit;
  527.             break;
  528.  
  529.         case TSYNC:                        /* Looks like an FTSC   */
  530.  
  531.             CLEAR_OUTBOUND ();            /* End noisy banner now */
  532.             (void) TIMED_READ (0);        /* Eat the TSYNC        */
  533.  
  534.             if (!no_WaZOO_Session)
  535.             {                            /* If we support WaZOO, */
  536.                 ret = RS6;                /* Go set up the timer  */
  537.             }
  538.             else
  539.             {
  540.                 args->result = 2;        /* FTS-0001 session     */
  541.                 ret = RSexit;
  542.             }
  543.  
  544.             break;
  545.  
  546.         case CR:                        /* Still sync'ing       */
  547.  
  548.             (void) TIMED_READ (0);        /* Eat the character    */
  549.             ret = RS4;                    /* Back to that stuff   */
  550.             break;
  551.  
  552.         case ESC:                        /* User wants the BBS   */
  553.  
  554.             (void) TIMED_READ (0);        /* Eat the character    */
  555.             if (!mail_only)
  556.             {
  557.                 args->result = 1;        /* BBS session          */
  558.                 ret = RSexit;
  559.             }
  560.             break;
  561.  
  562.         case '*':                        /* Looks like a EMSI    */
  563.  
  564.             if (!no_EMSI_Session)
  565.             {                            /* If we're EMSI,       */
  566.                 CLEAR_OUTBOUND ();        /* End noisy banner now */
  567.                 ret = RS8;
  568.                 break;
  569.             }
  570.  
  571.             /* Fall through if not EMSI */
  572.  
  573.         default:                        /* A less special char  */
  574.             if (i == -1)
  575.                 time_release ();        /*PLF 12-1-91  04:21:14 */
  576.             else
  577.             {
  578.                 if (args->emsi_flag)
  579.                     ret = RS8;
  580.                 else
  581.                     ret = RS7;            /* Check ext mailers    */
  582.             }
  583.             break;
  584.  
  585.         }
  586.         if (ret >= 0)
  587.             return (ret);
  588.  
  589.         if ((args->TSYNC_Timer != 0L) && (timeup (args->TSYNC_Timer)))
  590.         {
  591.             args->result = 2;            /* FTS-0001 session     */
  592.             return (RSexit);
  593.         }
  594.  
  595.         if (timeup (args->BBS_Timer))
  596.         {                                /* The big timeout      */
  597.             args->result = 1;            /* BBS session          */
  598.             return (RSexit);
  599.         }
  600.  
  601.         time_release ();                /* Give up the slice    */
  602.  
  603.     }                                    /* End (while CARRIER) */
  604.  
  605.     args->result = 0;                    /* carrier loss = abort */
  606.     return (RSexit);
  607. }
  608.  
  609. /*
  610.  
  611.  .-----+----------+-------------------------+-------------------------+-----.
  612.  | RS6 | TsyncTmr | 1. Timer not running    | Start 10 second timer   | RS5 |
  613.  |     |          |                         | Reset 20 sec timer      |     |
  614.  |     |          +-------------------------+-------------------------+-----|
  615.  |     |          | 2. Timer running        | Two TSYNCS = FTS-0001   | exit|
  616.  `-----+----------+-------------------------+-------------------------+-----'
  617.  
  618.  */
  619.  
  620. int 
  621. RSTsyncTmr (RSARGSP args)
  622. {
  623.     if (args->TSYNC_Timer == 0L)
  624.     {                                            /* If not already set,  */
  625.         args->TSYNC_Timer = timerset (1000);    /* Then just set it     */
  626.         args->BBS_Timer = timerset (BBStimeout);/* Reset 20 sec timer   */
  627.         return (RS5);
  628.     }
  629.     args->result = 2;                            /* FTS-0001 session     */
  630.     return (RSexit);
  631. }
  632.  
  633. /*
  634.  
  635.  .-----+----------+-------------------------+-------------------------+-----.
  636.  | RS7 | ExtMail  | 1. String matched       | External mail selected  | exit|
  637.  |     |          +-------------------------+-------------------------+-----|
  638.  |     |          | 2. Checked all strings  | Get next input character| RS5 |
  639.  `-----+----------+-------------------------+-------------------------+-----'
  640.  
  641.  */
  642.  
  643. int 
  644. RSExtMail (RSARGSP args)
  645. {
  646.     short i, k;
  647.  
  648.     /* Note: last character was read in RS5 using a NON-DESTRUCTIVE READ.
  649.       We now will get it out with a DESTRUCTIVE READ. */
  650.  
  651.     i = TIMED_READ (0) & 0xff;                        /* Eat the character     */
  652.  
  653.     for (k = 0; k < num_ext_mail; k++)
  654.     {
  655.         if (i != ((short) *(args->ExtMailScan[k]++) & 0xff))
  656.         {                                            /* Does this match next? */
  657.             args->ExtMailScan[k] = ext_mail_string[k];    /* No, reset pointer */
  658.         }
  659.         else
  660.         {
  661.             if (!*args->ExtMailScan[k])
  662.             {                                        /* End of UUCP string?   */
  663.                 args->result = 5 + k;                /* 5 + n for external    */
  664.                 return (RSexit);
  665.             }
  666.         }
  667.     }
  668.  
  669.     return (RS5);
  670. }
  671.  
  672. /*
  673.  
  674.  .-----+----------+-------------------------+-------------------------+-----.
  675.  | RS8 | EMSI     | 1. String matched       | Call EMSI handshake     | exit|
  676.  |     |          +-------------------------+-------------------------+-----|
  677.  |     |          | 2. No match             | Get next input character| RS5 |
  678.  `-----+----------+-------------------------+-------------------------+-----'
  679.  
  680.  */
  681.  
  682. int 
  683. RSEMSIChk (RSARGSP args)
  684. {
  685.     short i;
  686.  
  687.     /* Note: last character was read in RS5 using a NON-DESTRUCTIVE READ.
  688.       We now will get it out with a DESTRUCTIVE READ. */
  689.  
  690.     args->emsi_flag = TRUE;
  691.  
  692.     i = TIMED_READ (0);                                /* Eat the character     */
  693.     i = toupper (i & 0xff);
  694.  
  695.     if ((i != (short) (*(args->emsi_ptr))) &&
  696.         (i != (short) (*(args->iemsi_ptr))))
  697.     {                                                /* Does this match next? */
  698.         args->emsi_ptr = emsistr[EMSI_INQ];            /* No, reset pointer     */
  699.         args->iemsi_ptr = emsistr[EMSI_CLI];
  700.         args->emsi_flag = FALSE;
  701.     }
  702.     else
  703.     {
  704.         if (i == (short) (*(args->emsi_ptr)))
  705.             args->emsi_ptr++;
  706.         if (!*args->emsi_ptr)
  707.         {                                            /* End of EMSI string?   */
  708.             args->result = 4;                        /* 4 = EMSI */
  709.             return (RSexit);
  710.         }
  711.         if (i == (short) (*(args->iemsi_ptr)))
  712.             args->iemsi_ptr++;
  713.         if (!*args->iemsi_ptr && !mail_only)
  714.         {                                            /* End of EMSI string?   */
  715.             args->result = 1;                        /* 1 = BBS, fast IEMSI */
  716.             return (RSexit);
  717.         }
  718.     }
  719.     return (RS5);
  720. }
  721.